-
Notifications
You must be signed in to change notification settings - Fork 139
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
full-page templates #253
full-page templates #253
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What about if we just did it in JavaScript? The config.js could expose a template
function, and the default could be the one we have here.
This comment was marked as resolved.
This comment was marked as resolved.
PTAL |
@@ -0,0 +1,150 @@ | |||
import {parseHTML} from "linkedom"; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A bit of churn is due to moving the default rendering function and its utilities into a separate page.ts. I was (pleasantly) surprised that parseHtml is only needed for the toc.
* for pages in a subdirectory, etc. | ||
*/ | ||
root?: string; | ||
}, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What if we instead passed a declarative representation of the page state — rather than partially rendered HTML fragments — so that the template has complete flexibility in turning the declarative representation of the page into HTML (or anything else, e.g., TeX)? Though that requires us to design an intermediate representation as part of the public API. Worth the effort? 🤔
I also think we should encourage people to use the Html
helper here because any template implementation will be prone to escaping bugs without it. (The Html
helper is not Hypertext Literal (htl) — htl runs in the browser.)
I suspect we will also need to have an API for the CLI, declare a main
or exports
in package.json
, and probably run tsc
to convert TypeScript to JavaScript so that people can import helpers to implement their template function.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My approach here is very bare-bones, "stdout"-like. It treats observable markdown as a function that returns HTML fragments that you can just paste together and send as one big string to the browser for a severely unstyled but functional experience:
return title + preloads + module + main
All these fragments are made of safe HTML (unless I made a mistake). Users can assemble them in their own "page templates" (or placeholders) with zero difficulty. Pareto is happy.
A (crazy, or "pure") idea would be to return all of this as a single html fragment, and let the default template collect the elements with a querySelector (it already does it to build the TOC, so it wouldn't be much more costly). But probably too crazy.
Some elements are not HTML (path, root and the front-matter data); maybe we should put them in a separate argument?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@mbostock could you explain your declarative representation of the page state idea?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes… I need more time to think about this one and then come up with a more actionable proposal.
options: Omit<Config, "template"> | ||
) => string; | ||
|
||
export const page: Template = ({path, data, preloads, module, main, title, base, root}, options) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
looks like we want to eventually add other parameters for overriding other HTML fragments - like footer
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yep, no problem!
${title}${base}<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin> | ||
<link rel="stylesheet" type="text/css" href="https://fonts.googleapis.com/css2?family=Source+Serif+Pro:ital,wght@0,400;0,600;0,700;1,400;1,600;1,700&display=swap"> | ||
<link rel="stylesheet" type="text/css" href="${root}_observablehq/style.css">${ | ||
data?.template ? `\n<link rel="stylesheet" type="text/css" href="${root}_observablehq/${data.template}.css">` : "" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
data in this case comes directly from the front matter but what if the user wants to apply the same styling across all pages? maybe I am missing something...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree we'll want to allow customization of the global CSS (adding a stylesheet, or modifying the base stylesheet), but it's not in this PR's scope.
Full-page templates as a custom user function. The API is based on simple strings (no dependency on
Html
). The default page template computes the sidebar navigation and the pager.(This will also close #60 if we provide a dashboard.css.)